home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Menus / eudoraMenu.tcl < prev    next >
Encoding:
Text File  |  1998-08-21  |  24.6 KB  |  903 lines  |  [TEXT/ALFA]

  1. #     <nowrap>
  2. # Built for Eudora
  3.  
  4. # version 1.1 Juan Falgueras (98/08/07)
  5. #   Added eudoraNicknamesFolder pref var to hold the path to the Eudora "Nicknames Folder"
  6. #     and added this folder in the scanning for "euNicknames" array (look for # JF)
  7. # version 1.2 Juan Falgueras (98/8/21)
  8. #   Rewrite of finishNickname using prompt::statusLineComplete and deleting substituteNickname
  9. # TO DO
  10. #   Take the folders (Not ended in " Folder") from the Eudora Folder (recursively) to
  11. #   build the MailBoxes.  Not ask for it to Eudora via AE.
  12. #     
  13. # Class message: A message
  14. # Elements:
  15. #     field by name
  16. # Properties:
  17. #     body  string  -- the body of the message
  18. #     priority  integer  -- the priority
  19. #     label  integer  -- the index of the label
  20. #     status  unread/already read/replied/forwarded/redirected/not sendable/sendable/queued/sent/never sent  -- the message status
  21. #     sender  string  -- the sender as appearing in the message summary
  22. #     date  string  [r/o]  -- the date as appearing in the message summary
  23. #     subject  string  -- the subject as appearing in the message summary
  24. #     size  integer  [r/o]  -- the size of the message
  25. #     outgoing  boolean  [r/o]  -- is the message is outgoing?
  26. #     signature  none/standard/alternate  -- which signature the message should have
  27. #     QP  boolean  -- is Eudora allowed to encode text?
  28. #     return receipt  boolean  -- is a return receipt is requested?
  29. #     wrap  boolean  -- should the text be wrapped when sent?
  30. #     tab expansion  boolean  -- should tabs get expanded to spaces?
  31. #     keep copy  boolean  -- should a copy should be kept after message is sent?
  32. #     preserve macintosh info  boolean  -- should Macintosh information always be sent with attachments?
  33. #     attachment encoding  AppleDouble/AppleSingle/BinHex/uuencode  -- the type of encoding to use for attachments
  34. #     show all headers  boolean  -- should all headers be visible?
  35. #     transliteration table  integer  -- the resource id of the transliteration table
  36. #     will be fetched  boolean  -- will the message be [re]fetched on next check?
  37. #     will be deleted  boolean  -- will the message be deleted from server on next check?
  38.  
  39. alpha::menu eudoraMenu 0.1 "•138" {global Mail} {
  40.     addMode Mail dummyMail {} {eudoraMenu}
  41.     set unixMode(rmail) {Mail}
  42. } uninstall {this-file} help {file "Eudora"}
  43.  
  44. proc eudoraMenu {} {}
  45. proc dummyMail {} {}
  46.  
  47. namespace eval ::Mail {}
  48.  
  49. if {![info exists trashName]} {set trashName "Trash"}
  50.  
  51. newPref v prefixString {> } Mail
  52. newPref v wordBreak {\w+} Mail
  53. newPref v wordBreakPreface {(\W)} Mail
  54. newPref f wordWrap {1} Mail
  55. newPref f autoMark 0 Mail
  56. newPref v eudoraPrefFolder  [file join [file dirname $PREFS] "Eudora Folder"] Mail
  57. newPref file eudoraNicknames "" Mail
  58. newPref v eudoraNicknamesFolder "" Mail
  59. newPref f tossOnQueue 1 Mail
  60. newPref f switchOnQueue 0 Mail
  61.  
  62. set mailKeywords {{Subject:} {To:} {From:} {Cc:} {Date:} {Sender:}}
  63. regModeKeywords -k blue -e {>} Mail $mailKeywords
  64.  
  65. # bind tab
  66. Bind 0x30   mailTab Mail
  67.  
  68. #===============================================================================
  69. # Utility routines.
  70. proc eudoraFolder {} { nameObject euMF {'TEXT'()} [nullObject] }
  71. proc mailboxByName {name} { nameObject euMB "“$name”" [eudoraFolder] }
  72. proc mailboxByIndex {ind} { indexObject euMB $ind [eudoraFolder] }
  73. proc eudoraMessage {msg_id mailbox} { indexObject euMS $msg_id [mailboxByName $mailbox] }
  74. proc mailboxProperty {prop mailbox} { objectProperty 'CSOm' $prop [mailboxByName $mailbox] }
  75. proc messageProperty {prop msg_id mailbox} { objectProperty 'CSOm' $prop [eudoraMessage $msg_id $mailbox] }
  76.  
  77. # Get path-name for indicated mailbox
  78. proc mailboxPathName {name} {
  79.     extractPath [mailboxProperty euFS $name]
  80. }
  81.  
  82. proc mailboxPathIndex {ind} {
  83.     set res [objectProperty 'CSOm' euFS [mailboxByIndex $ind]]
  84.     return [extractPath $res]
  85. }
  86.  
  87. #===============================================================================
  88.  
  89.  
  90. proc checkMailPath {} {
  91.     set name [nameFromAppl CSOm]
  92.     launch $name
  93.     return [file tail $name]
  94. }
  95.  
  96. if {![info exists eudoraBoxes]} {
  97.     set eudoraBoxes {In Out}
  98. }
  99.  
  100. Menu -n $eudoraMenu -p eudoraProc {
  101.     "<SbackgroundEudora"
  102.     "<Seudora"
  103.     "help"
  104.     "(-"
  105.     "/e<U<BnewMessage"
  106.     "cc"
  107.     "bcc"
  108.     "(-"
  109.     "/f<U<Bsend"
  110.     "/g<U<S<Breply"
  111.     "/g<U<I<S<BreplyToAll"
  112.     "(-"
  113.     "/,<SfinishNickname"
  114.     "editNicknames"
  115.     "updateNicknames"
  116.     "(-"
  117.     "updateMailboxLists"
  118.     {Menu -n open -p mailBoxProc -m {}}
  119.     {Menu -n moveTo -p transferProc -m {}}
  120.     "(-"
  121.     {Menu -n tellEudora -p eudoraProc {
  122.         "/k<UflushOutbox"
  123.         "/l<UcheckForMail"
  124.         "(-"
  125.         startNotifying
  126.         stopNotifying}}
  127.     {Menu -n mailFlags -p mailFlagsProc {alertOnIncoming flushOnCheck immediateSend -- trashName}}
  128. }
  129. Menu -n open -p mailBoxProc -m [concat [list "/h<U<BOpen Mailbox…" "(-"] $eudoraBoxes]
  130. Menu -n moveTo -p transferProc -m [concat [list "/i<U<BMove To Trash" "/j<U<BMove To Mailbox…" "(-"] $eudoraBoxes]
  131.  
  132. # JF 98/08/07
  133. if {$MailmodeVars(eudoraNicknames) == ""} {
  134.     if [file exists "$MailmodeVars(eudoraPrefFolder):Eudora Nicknames"] {
  135.         set MailmodeVars(eudoraNicknames) "$MailmodeVars(eudoraPrefFolder):Eudora Nicknames"
  136.     }
  137. }
  138. if {$MailmodeVars(eudoraNicknamesFolder) == ""} {
  139.     if [file exists "$MailmodeVars(eudoraPrefFolder):Nicknames Folder"] {
  140.         set MailmodeVars(eudoraNicknamesFolder) "$MailmodeVars(eudoraPrefFolder):Nicknames Folder"
  141.     }
  142. }
  143.  
  144.  
  145.  
  146. proc updateNicknames {{arg ""}} {
  147.     global euNicknames MailmodeVars
  148. # JF 98/08/07
  149.     if {(![info exists MailmodeVars(eudoraNicknames)] && ![info exists MailmodeVars(eudoraNicknamesFolder)])\
  150.                   || \
  151.       (![file exists $MailmodeVars(eudoraNicknames)] && ![file exists $MailmodeVars(eudoraNicknamesFolder)])} {
  152.         if {![string length $arg]} {
  153.             alertnote {Please locate the file "Eudora Nicknames" or/and folder "Nicknames Folder" using Mail mode's prefs.}
  154.         }
  155.         return
  156.     }
  157.     
  158.     if [file exists $MailmodeVars(eudoraNicknames)] {
  159.         set fd [open $MailmodeVars(eudoraNicknames)]
  160.         foreach a [split [read $fd] "\n"] {
  161.             if {[llength $a] && [lindex $a 0] == "alias"} {
  162.                 set euNicknames([lindex $a 1]) [lindex $a 2]
  163.             }
  164.         }
  165.         close $fd
  166.     }
  167.     if [expr {$MailmodeVars(eudoraNicknamesFolder) != ""} && [file isdirectory $MailmodeVars(eudoraNicknamesFolder)]] {
  168.         foreach f [glob "$MailmodeVars(eudoraNicknamesFolder):*"] {
  169.             set fd [open "$f"]
  170.             foreach a [split [read $fd] "\n"] {
  171.                 if {[regexp {\{\}\[\]} $a]} {
  172.                     alertnote "ERROR: Braces in “$f”"
  173.                     close $fd
  174.                     return
  175.                 } else {
  176.                     if {[llength $a] && [lindex $a 0] == "alias"} {
  177.                         set euNicknames([lindex $a 1]) [lindex $a 2]
  178.                     }
  179.                 }
  180.             }
  181.             close $fd
  182.         }
  183.     }
  184. }
  185. updateNicknames quiet
  186.  
  187.  
  188. proc finishNickname {} {
  189. #     I don't know well the mechanism of prompt::statusLineComplete and I needed
  190. #     to use a global :( preMailNick to prefix the previous entered string before each
  191. #     search for matchings in nicknames
  192. # Another interesting thing should be to show the initial matched list just at
  193. #     starting of the command.  You need now to add a char to start to see the list
  194.  
  195.     global euNicknames preMailNick
  196.     set pos [getPos]
  197.     backwardWord
  198.     if {[getPos] == 0} {
  199.         set preMailNick ""
  200.     } else {
  201.         set preMailNick [getText [getPos] $pos]
  202.     }
  203.  
  204.     goto $pos
  205.     ;proc pMatches {pat} {
  206.         global euNicknames preMailNick
  207.         set matches ""
  208.         foreach w [array names euNicknames] {
  209.             if {[string match "$preMailNick$pat*" $w]} {
  210.                 lappend matches $w
  211.             }
  212.         }
  213.         return $matches
  214.     }
  215.     
  216.     set s [prompt::statusLineComplete "Nick" pMatches -initialpatt "$preMailNick"]
  217.     if {$preMailNick != ""} {
  218.         backwardDeleteWord
  219.     }
  220.     insertText $s
  221. }
  222.  
  223. proc editNicknames {} {
  224.     global MailmodeVars
  225. # JF 98/08/07
  226.     if {![info exists MailmodeVars(eudoraNicknamesFolder)]} {
  227.         edit -w "$MailmodeVars(eudoraNicknames)"
  228.     } else {
  229.         edit -w [getfile "Which of them?" "$MailmodeVars(eudoraNicknamesFolder)"]
  230.     }
  231. }
  232.  
  233.  
  234. proc openMailbox {} {
  235.     global eudoraBoxes eudoraLastFolder modifiedVars
  236.     if {[info exists eudoraLastFolder]} {
  237.         set fold $eudoraLastFolder
  238.     } else {
  239.         set fold [lindex $eudoraBoxes 0]
  240.     }
  241.     set eudoraLastFolder [sPromptChoices "Open mailbox" $fold $eudoraBoxes]
  242.     mailBoxProc dummy $eudoraLastFolder
  243.     lappend modifiedVars eudoraLastFolder
  244. }
  245.  
  246.  
  247. proc moveToFolderProc {curr c} {
  248.     global eudoraBoxes
  249.     if {$c != "\t"} {return $c}
  250.     
  251.     set matches {}
  252.     foreach w $eudoraBoxes {
  253.         if {[string match "$curr*" $w]} {
  254.             lappend matches $w
  255.         }
  256.     }
  257.     if {![llength $matches]} {
  258.         beep
  259.     } else {
  260.         return [string range [largestPrefix $matches] [string length $curr] end]
  261.     }
  262.     return ""
  263. }
  264.  
  265.  
  266. proc moveToMailbox {} {
  267.     global eudoraLastFolder modifiedVars trashName
  268.     
  269.     if {[info exists eudoraLastFolder]} {
  270.         set fold $eudoraLastFolder
  271.     } else {
  272.         set fold $trashName
  273.     }
  274.     
  275.     set folder [statusPrompt -f "Move to ($fold): " moveToFolderProc]
  276.     if {[string length $folder]} {
  277.         doFolderMove $folder
  278.         set eudoraLastFolder $folder
  279.         lappend modifiedVars eudoraLastFolder
  280.     } else {
  281.         doFolderMove $fold
  282.     }
  283. }
  284.  
  285.  
  286. proc eudoraProc {menu item} {
  287.     global HOME eudoraMenu
  288.  
  289.     switch -- $item {
  290.         "eudora"        {app::launchFore CSOm}
  291.         "backgroundEudora"        {launch [nameFromAppl CSOm]}
  292.         "help"            {edit -r [file join $HOME Help Eudora]}
  293.         "cc"            {eudoraCc}
  294.         "bcc"            {eudoraBcc}
  295.         "newMessage"    {mailNewMsg}
  296.         "reply"            {mailReplymsg}
  297.         "replyToAll"    {mailReplymsg 1}
  298.         "send"            {checkMailPath; mailSendCreatedMsg}
  299.         "flushOutbox"    {checkMailPath; mailFlushOut}
  300.         "checkForMail"    {checkMailPath; mailCheck}
  301.         "startNotifying" "checkMailPath; startNotifying"
  302.         "stopNotifying"    "checkMailPath; stopNotifying"
  303.         "moveToTrash"    {mailTrashmsg}
  304.         "Move To Trash"    {mailTrashmsg}
  305.         "Create New"    {eudoraNewMailbox}
  306.         "Delete"        {eudoraDeleteMailbox}
  307.         default            $item
  308.     }
  309. }
  310.  
  311. if {![info exists mailflushOnCheck]} {set mailflushOnCheck 1}
  312. if {![info exists mailalertOnIncoming]} {set mailalertOnIncoming 1}
  313. if {![info exists mailimmediateSend]} {set mailimmediateSend 0}
  314. markMenuItem mailFlags flushOnCheck $mailflushOnCheck
  315. markMenuItem mailFlags alertOnIncoming $mailalertOnIncoming
  316. markMenuItem mailFlags immediateSend $mailimmediateSend
  317.  
  318.  
  319. proc mailFlagsProc {menu flag} {
  320.     global mail$flag modifiedVars trashName
  321.  
  322.     if {$flag == "trashName"} {
  323.         set trashName [prompt "Trash folder name:" $trashName]
  324.         lappend modifiedVars trashName
  325.         return
  326.     }
  327.     
  328.  
  329.     set mail$flag [expr 1 - [set mail$flag]]
  330.     lappend modifiedVars mail$flag
  331.     markMenuItem mailFlags $flag [set mail$flag]
  332. }
  333.  
  334.  
  335. proc mailBoxProc {menu item} {
  336.     global inboxMembers ALPHA eudoraBoxes modifiedVars
  337.     global tileLeft tileTop tileHeight errorHeight defWidth
  338.  
  339.     if {$item == "Open Mailbox"} {
  340.         return [openMailbox]
  341.     }
  342.     
  343.     killMailboxWindow
  344.  
  345.     checkMailPath
  346.     switchTo $ALPHA
  347.  
  348.     global trashedMsgs$item
  349.     set trashedMsgs$item {}
  350.     
  351.     set inboxMembers {}
  352.     set text {}
  353.     set ind 1
  354.     foreach msg [mailSenders $item] {
  355.         set from [mailAddr [lindex $msg 0]]
  356.         set tag {}
  357.         while {[lsearch $inboxMembers $from$tag] >= 0} {
  358.             if {![string length $tag]} {
  359.                 set tag { <2>}
  360.             } else {
  361.                 regexp {[0-9]+} $tag tag
  362.                 set tag " <[expr $tag + 1]>"
  363.             }
  364.         }
  365.         append text [format "%-40s : %s\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t∞$item∞$ind\r" "$from$tag" [lindex $msg 1]]
  366.         lappend inboxMembers $from$tag
  367.         incr ind
  368.     }
  369.     if {![string length $text]} {
  370.         alertnote "No messages in '$item'!"
  371.         return
  372.     }
  373.     
  374.     new -n "* MAILBOX '$item' *" -g $tileLeft $tileTop $defWidth $errorHeight -m Brws
  375.     insertText "(<cr> to go to message)\r-----\r$text"
  376.     
  377.     select [nextLineStart [nextLineStart [minPos]]] [nextLineStart [nextLineStart [nextLineStart [minPos]]]]
  378.     winReadOnly
  379.     message ""
  380. }
  381.  
  382. proc eudoraNewMailbox {} {
  383. }
  384.  
  385.  
  386. proc eudoraDeleteMailbox {} {
  387. }
  388.  
  389.  
  390. proc killMailboxWindow {} {
  391.     if {[set ind [lsearch [winNames] {*MAILBOX*}]] >= 0} {
  392.         set win [lindex [winNames] $ind]
  393.         bringToFront $win
  394.         killWindow
  395.     }
  396. }
  397.  
  398.  
  399. proc mailGotoMatch {} {
  400.     if {[regexp {∞(.+)∞([0-9]+)} [getText [lineStart [getPos]] [nextLineStart [getPos]]] dummy folder ind]} {
  401.         global trashedMsgs$folder
  402.         
  403.         set deleted 0
  404.         for {set i 1} {$i < $ind} {incr i} {
  405.             if {[lsearch [set trashedMsgs$folder] $i] >= 0} {
  406.                 incr deleted
  407.             }
  408.         }
  409.         mailOpen $folder [expr $ind - $deleted] $ind
  410.     }
  411. }
  412.  
  413.  
  414. proc mailAddr {name} {
  415.     if {![regexp {<(.*)>} $name dummy addr]} {
  416.         regexp {^[^ ]+} $name addr
  417.     }
  418.     return $addr
  419. }
  420.  
  421.  
  422. proc mailSenders {folder} {
  423.     set cnt [mailCountMsgs $folder]
  424.     set msgs {}
  425.     for {set i 1} {$i <= $cnt} {incr i} {
  426.         set subject {}
  427.         if {![regexp -nocase {From: (.*)} [mailGetField from $folder $i] dummy from]} {
  428.             error "No from field!"
  429.         }
  430.         regexp {Subject: (.*)} [mailGetField subject $folder $i] dummy subject
  431.         lappend msgs [list $from $subject]
  432.     }
  433.     return $msgs
  434. }    
  435.  
  436.  
  437. proc eudoraCc {} {
  438.     if {![catch {set res [search -s -f 1 -r 1 -i 1 {cc:.*} 0]}]} {
  439.         goto [lindex $res 0]
  440.         endOfLine
  441.     } else {
  442.         beginningOfBuffer
  443.         nextLine
  444.         insertText "Cc: \r"
  445.         backwardChar
  446.     }
  447. }
  448.  
  449.  
  450. proc eudoraBcc {} {
  451.     if {![catch {set res [search -s -f 1 -r 1 -i 1 {bcc:.*} 0]}]} {
  452.         goto [lindex $res 0]
  453.         endOfLine
  454.     } else {
  455.         beginningOfBuffer
  456.         nextLine
  457.         insertText "Bcc: \r"
  458.         backwardChar
  459.     }
  460. }
  461.  
  462.  
  463. proc mailFlushOut {} {
  464.     global MailmodeVars
  465.     message "Telling Eudora to flush messages…"
  466.  
  467.     AEBuild 'CSOm' CSOm eCon eSen bool(«01») eChk bool(«00»)
  468.     message ""
  469. }
  470.  
  471.  
  472. proc mailCheck {} {
  473.     global MailmodeVars mailflushOnCheck
  474.     message "Told Eudora to check for new mail…"
  475.  
  476.     AEBuild 'CSOm' CSOm eCon eSen bool(«0$mailflushOnCheck») eChk bool(«01»)
  477. }
  478.  
  479.  
  480. proc mailReplymsg {{toall 0}} {
  481.     global inboxMembers
  482.  
  483.     set prefix "> "
  484.     set res [search -s -f 1 -r 1 {^>==} [minPos]]
  485.     set header [getText [minPos] [lindex $res 0]]
  486.     
  487.     set from [mailGetFrom]
  488.     if {![regexp {Subject: ([^\r]*)} $header dummy subject]} {set subject ""}
  489.     set body [getText [expr [lindex $res 1] + 1] [maxPos]]
  490.  
  491.     regsub -all "\r" $body "\r$prefix" body
  492.     mailNewMsg $from "" "" "Re: $subject" "\r$prefix$body\r"
  493. }
  494.  
  495. proc transferProc {menu item} {
  496.     if {$item == "Move To Trash"} {
  497.         mailTrashmsg
  498.     } elseif {$item == "Move To Mailbox"} {
  499.         moveToMailbox
  500.     } else {
  501.         doFolderMove $item
  502.     }
  503. }
  504.  
  505.  
  506. proc mailTrashmsg {} {
  507.     global trashName
  508.     doFolderMove $trashName
  509. }
  510.  
  511. proc doFolderMove {toFolder} {
  512.     # Is this a summary or msg window?
  513.     if {[regexp "MAILBOX" [win::CurrentTail]]} {
  514.         if {![regexp {∞(.*)∞([0-9]+)} [getText [getPos] [nextLineStart [getPos]]] dummy folder orig]} {
  515.             beep
  516.             return
  517.         }
  518.         if {$toFolder == $folder} return
  519.         global trashedMsgs$folder
  520.  
  521.         set deleted 0
  522.         for {set i 1} {$i < $orig} {incr i} {
  523.             if {[lsearch [set trashedMsgs$folder] $i] >= 0} {
  524.                 incr deleted
  525.             }
  526.         }
  527.         set number [expr $orig - $deleted]
  528.         set summary 1
  529.     } else {
  530.         set pos [lindex [search -f 1 -r 1 {^Msg} 0] 0]
  531.         set text [getText $pos [nextLineStart $pos]]
  532.         regexp {"([^"]+)" \(([0-9]+)\).*"([^"]+)"} $text dummy number orig folder
  533.         if {$toFolder == $folder} return
  534.         set summary 0
  535.  
  536.         global trashedMsgs$folder
  537.     }
  538.     
  539.     message "Moving msg $number ($orig) of folder '$folder' to '$toFolder'"
  540.     moveMsg $number $folder $toFolder
  541.     if {!$summary} killWindow
  542.  
  543.     lappend trashedMsgs$folder $orig
  544.     
  545.     # Find summary info and delete it
  546.     set win [win::CurrentTail]
  547.     if {[regexp "MAILBOX" $win]} {
  548.         setWinInfo read-only 0
  549.     
  550.         set inds [search -f 1 -r 1 "∞$folder∞$orig\$" 0]
  551.         set pos [lindex $inds 0]
  552.         deleteText [lineStart $pos] [nextLineStart $pos]
  553.     
  554.         if {[string length [search -n -f 1 -r 0 {∞} 0]]} {
  555.             setWinInfo dirty 0
  556.             setWinInfo read-only 1
  557.     
  558.             nextLine
  559.             upBrowse
  560.         } else {
  561.             setWinInfo dirty 0
  562.             killWindow
  563.         }
  564.     }
  565. }
  566.  
  567.     
  568. proc mailGetFrom {} {
  569.     set res [search -s -f 1 -r 1 {>==} [minPos]]
  570.     set header [getText [minPos] [lindex $res 0]]
  571.     if {[regexp {From: ([^\r]*)} $header dummy from]} {
  572.         return [mailAddr $from]
  573.     }
  574.     error "No from line"
  575. }
  576.  
  577.     
  578.  
  579. proc mailSendCreatedMsg {} {
  580.     global MailmodeVars mailimmediateSend
  581.     
  582.     set name [checkMailPath]
  583.     set res [search -s -f 1 -r 0 {>==text follows this line==<} [minPos]]
  584.     set header [getText [minPos] [lindex $res 0]]
  585.     
  586.     if {![regexp {To: ([^\r]*)} $header dummy to]} {set to ""}
  587.     if {![regexp {[Cc]+: ([^\r]*)} $header dummy cc]} {set cc ""}
  588.     if {![regexp {Subject: ([^\r]*)} $header dummy subject]} {set subject ""}
  589.  
  590.     set body [getText [expr [lindex $res 1] + 1] [maxPos]]
  591.     
  592.     mailCreateMsg
  593.     mailSetField to $to
  594.     mailSetField cc $cc
  595.     mailSetField subject $subject
  596.     mailSetField "" $body
  597.     
  598.     if {$MailmodeVars(tossOnQueue)} {
  599.         setWinInfo dirty 0
  600.         killWindow
  601.     }
  602.  
  603.     mailQueueLast
  604.  
  605.     if {$MailmodeVars(switchOnQueue)} {
  606.         switchTo $name
  607.     }
  608.  
  609.     if {$mailimmediateSend} {
  610.         mailFlushOut
  611.     } else {
  612.         message "Message queued…"
  613.     }
  614. }
  615.  
  616.     
  617. proc mailNewMsg {{to ""} {cc ""} {bcc ""} {subject ""} {body ""}} {
  618.     new -n "New Mail" -m Mail
  619.  
  620.     insertText "To: $to\rSubject: $subject\r>==text follows this line==<\r$body"
  621.     setWinInfo dirty 0
  622.     goto [minPos]
  623.     refresh
  624.     mailTab
  625. }
  626.  
  627.  
  628. proc mailTab {} {
  629.     global mailKeywords
  630.     if {[catch {search -s -f 1 -r 1 -i 1 "([join $mailKeywords {|}]|^>)" [getPos]} res]} {
  631.         insertText "\t"
  632.         return
  633.     }
  634.     if {[lookAt [lindex $res 0]] == ">"} {
  635.         select [nextLineStart [lindex $res 1]] [maxPos]
  636.     } else {
  637.         goto [expr [lindex $res 1] + 1]
  638.         endLineSelect
  639.     }
  640. }
  641.  
  642. proc mailOpen {folder i originalNum} {
  643.     global tileLeft tileTop tileHeight trashedMsgs$folder defWidth
  644.     
  645.     set from [mailGetField From $folder $i]
  646.     set to [mailGetField To $folder $i]
  647.     set subject [mailGetField Subject $folder $i]
  648.     set sender [mailGetField Sender $folder $i]
  649.     set date [mailGetField Date $folder $i]
  650.     set cc [mailGetField cc $folder $i]
  651.     set contents [mailGetField "" $folder $i]
  652.     
  653. #===============================================================================
  654.     global tileHeight tileTop tileLeft errorHeight errorDisp defWidth
  655.     set top $tileTop
  656.     set geo [getGeometry]
  657.     if {([lindex $geo 0] != $tileLeft) || ([lindex $geo 1] != $top) || ([lindex $geo 2] != $defWidth) || ([lindex $geo 3] != $errorHeight) } {
  658.         moveWin $tileLeft $top
  659.         sizeWin $defWidth $errorHeight
  660.     }
  661.     set mar 28
  662.     incr top [expr $errorHeight + $mar]
  663. #===============================================================================
  664.     set name [new -n "MAIL" -g $tileLeft $top $defWidth [expr $errorDisp - 5] -m Mail]
  665.  
  666.     set text {}
  667.     if {[string length $subject]} {append text "$subject\r"}
  668.     if {[string length $to]} {append text "$to\r"}
  669.     if {[string length $from]} {append text "$from\r"}
  670.     if {[string length $date]} {append text "$date\r"}
  671.     if {[string length $cc]} {append text "$cc\r"}
  672.     if {[string length $sender]} {append text "$sender\r"}
  673.     append text "Msg \"$i\" ($originalNum) of mailbox \"$folder\"     Reply     Trash\r"
  674.     insertText "${text}>===============================================================================
  675. $contents
  676. "
  677.     goto [minPos]
  678.     goto [lindex [search -f 1 -m 0 -r 1 {^Msg "} [minPos]] 0]
  679.     set pos [getPos]
  680.     regexp -indices {([0-9]+).*mailbox "(.+)".*(Reply).*(Trash)} [getText $pos [nextLineStart $pos]] dummy num fold rep tra
  681.     colorSelectPos [expr $pos + [lindex $num 0]] [expr 1 + $pos + [lindex $num 1]] bold 12
  682.     colorSelectPos [expr $pos + [lindex $fold 0]] [expr 1 +$pos + [lindex $fold 1]] bold 12
  683.     hyperSelectPos [expr $pos + [lindex $rep 0]] [expr 1 +$pos + [lindex $rep 1]] "mailReplymsg"
  684.     hyperSelectPos [expr $pos + [lindex $tra 0]] [expr 1 +$pos + [lindex $tra 1]] "mailTrashmsg"
  685.  
  686.     winReadOnly
  687.     refresh
  688. }
  689.  
  690. proc colorSelectPos {from to ind ind2} {
  691.     insertColorEscape $from $ind
  692.     insertColorEscape $to $ind2
  693. }
  694.  
  695. proc hyperSelectPos {from to text} {
  696.     if {$from == $to} {
  697.         beep
  698.         return
  699.     }
  700.     insertColorEscape $from 3
  701.     insertColorEscape $from 15 $text
  702.     insertColorEscape $to 12
  703.     insertColorEscape $to 0
  704. }
  705.  
  706. proc mailFixMenus {} {
  707.     global eudoraMenu
  708.     
  709.     set ins {}
  710.     set outs {}
  711.     
  712.     set cnt [mailCountMsgs in]
  713.     for {set i 1} {$i <= $cnt} {incr i} {
  714.         lappend ins [mailGetField from in $i]
  715.     }
  716.     Menu -n in {}
  717.     foreach item $ins {
  718.         addMenuItem -l "" in $item
  719.     }
  720.     
  721.     set cnt [mailCountMsgs out]
  722.     for {set i 1} {$i <= $cnt} {incr i} {
  723.         lappend outs [mailGetField from out $i]
  724.     }
  725.     Menu -n out {}
  726.     foreach item $outs {
  727.         addMenuItem -l "" out $item
  728.     }
  729. }
  730.  
  731.  
  732. # make message at end of mailbox "out" of mail folder ""
  733. proc mailCreateMsg {} {
  734.     createThingAtEnd 'CSOm' [mailboxByName Out] euMS
  735. }
  736.  
  737.  
  738. # Move msg w/ specified index between folders, including to Trash.
  739. proc moveMsg {msg infolder outfolder} {
  740.     AEBuild -r 'CSOm' core move {----} "obj \{form:indx, want:type(euMS), seld:$msg, from:obj \{form:name, want:type(euMB), seld:“$infolder”, from:obj \{form:name, want:type(euMF), seld:'TEXT'(), from:'null'()\}\}\}" {insh} "insl\{kobj:obj \{form:name, want:type(euMB), seld:“$outfolder”, from:obj \{form:name, want:type(euMF), seld:'TEXT'(), from:'null'()\}\}, kpos:end \}"
  741. }
  742.  
  743.  
  744. # set field "fld" of message 0 to "to"
  745. proc mailSetField {fld to} {
  746.     AEBuild -r 'CSOm' core setd {----} "obj \{form:name, want:type(euFd), seld:“$fld”, from:obj \{form:indx, want:type(euMS), seld:1, from:'null'()\}\}" data "“$to”"
  747. }
  748.  
  749. proc mailGetField {field folder msg} {
  750.     if {[catch {getObjectData 'CSOm' euFd $field [eudoraMessage $msg $folder]} res]} {
  751.         return ""
  752.     } else {
  753.         return $res
  754.     }
  755. }
  756.  
  757. # obj {want:type('prop'), from:obj {form:indx, want:type(euMS), seld:$msg, from:obj {form:name, want:type(euMB), seld:“$folder”, from:obj {form:name, want:type(euMF), seld:'TEXT'(), from:'null'()}}}, form:'prop', seld:type('euST')}
  758.  
  759. proc mailMsgStatus {folder msg} {
  760.     if {[catch {AEBuild -r 'CSOm' core getd {----} "obj \{want:type('prop'), from:obj \{form:indx, want:type(euMS), seld:$msg, from:obj \{form:name, want:type(euMB), seld:“$folder”, from:obj \{form:name, want:type(euMF), seld:'TEXT'(), from:'null'()\}\}\}, form:'prop', seld:type('euST')\}"} text]} {
  761.         return ""
  762.     }
  763.     set from [string first "“" $text]
  764.     set to [string first "”" $text]
  765.     return [string range $text [expr $from + 1] [expr $to - 1]]
  766. }
  767.  
  768. proc mailQueueLast {} {
  769.     AEBuild -r 'CSOm' CSOm eQue ---- {obj {form:indx, want:type(euMS), seld:1, from:'null'()}}
  770. }
  771.  
  772. # CSOm\eAtc{'----':obj {form:indx, want:type(euMS), seld:1, from:'null'()}, eDcl:[fss («486172643A4465736B746F7020466F6C6465723A62756773»)]}
  773. proc mailAttachmentDoesntWork {name} {
  774.     AEBuild -r 'CSOm' CSOm eAtc ---- {obj {form:indx, want:type(euMS), seld:1, from:'null'()}} eDcl [makeAlis $name]
  775. }
  776.  
  777. # core\setd{'----':obj {form:prop, want:type(prop), seld:type(euPY), from:obj {form:indx, want:type(euMS), seld:1, from:'null'()}}, data:1}
  778. proc mailSetNumberProperty {prop to} {
  779.     AEBuild -r 'CSOm' core setd ---- "obj \{form:prop, want:type(prop), seld:type($prop), from:obj \{form:indx, want:type(euMS), seld:1, from:'null'()\}\}" data $to
  780. }    
  781.  
  782.  
  783. proc mailCountMsgs {mbox} {
  784.     return [countObjects 'CSOm' [mailboxByName "$mbox"]  euMS]
  785. }
  786.  
  787. proc mailCountMailboxes {} {
  788.      return [countObjects 'CSOm' [eudoraFolder] euMB]
  789. }
  790.     
  791.  
  792. proc updateMailboxLists {} {
  793.     set num [mailCountMailboxes]
  794.     for {set i 1} {$i <= $num} {incr i} {
  795.         set name [file tail [mailboxPathIndex $i]]
  796.         message $name
  797.         lappend boxes $name
  798.     }
  799.     
  800.     global eudoraBoxes modifiedVars
  801.     lappend modifiedVars eudoraBoxes
  802.     set eudoraBoxes $boxes
  803.     Menu -n open -p mailBoxProc -m $eudoraBoxes
  804.     Menu -n moveTo -p transferProc -m $eudoraBoxes
  805.     message "done."
  806. }
  807.  
  808.  
  809. #===============================================================================
  810.  
  811. eventHandler CSOm eNot "eudoraHandler"
  812.  
  813. # a sample 'msgs': obj {want:type(euMS), from:obj {want:type(euMB), from:obj {want:type(euMF), from:'null'(), form:name, seld:'TEXT'()}, form:name, seld:“In”}, form:indx, seld:18}, obj {want:type(euMS), from:obj {want:type(euMB), from:obj {want:type(euMF), from:'null'(), form:name, seld:'TEXT'()}, form:name, seld:“In”}, form:indx, seld:19}
  814. proc eudoraHandler {it} {
  815.     global blah mailalertOnIncoming
  816.     message ""
  817.     if {[regexp {eWHp:wArv.*\[(obj.*)\], &repq} $it dum1 msgs]} {
  818.         set ids [getMsgIDs $msgs]
  819.         if {$mailalertOnIncoming} {
  820.             mailBrowser In $ids
  821.         }
  822.     } else {
  823.         message "No mail"
  824.     }
  825. }
  826.  
  827.  
  828. proc mailBrowser {folder ids} {
  829.     global inboxMembers ALPHA
  830.     global tileLeft tileTop defWidth tileHeight errorHeight
  831.  
  832.     checkMailPath
  833.     switchTo $ALPHA
  834.  
  835.     global trashedMsgs$folder
  836.     set trashedMsgs$folder {}
  837.     
  838.     set inboxMembers {}
  839.     set text {}
  840.  
  841.     foreach id $ids {
  842.         regexp {From: (.*)} [mailGetField from $folder $id] dummy from
  843.         set from [mailAddr $from]
  844.         set subject {}
  845.         regexp {Subject: (.*)} [mailGetField subject $folder $id] dummy subject
  846.         set tag {}
  847.         while {[lsearch $inboxMembers $from$tag] >= 0} {
  848.             if {![string length $tag]} {
  849.                 set tag { <2>}
  850.             } else {
  851.                 regexp {[0-9]+} $tag tag
  852.                 set tag " <[expr $tag + 1]>"
  853.             }
  854.         }
  855.         append text [format "%-40s : %s\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t∞In∞$id\r" "$from$tag" $subject]
  856.         lappend inboxMembers $from$tag
  857.     }
  858.     if {![string length $text]} {
  859.         alertnote "No messages!"
  860.         return
  861.     }
  862.     
  863.     if {[set ind [lsearch [winNames] "*Incoming MAILBOX*"]] >= 0} {
  864.         bringToFront [lindex [winNames] $ind]
  865.         setWinInfo read-only 0
  866.         endOfBuffer
  867.         insertText $text
  868.     } else {
  869.         killMailboxWindow
  870.         new -n "* Incoming MAILBOX *" -g $tileLeft $tileTop $defWidth $errorHeight -m Brws
  871.  
  872.         insertText "(<cr> to go to message)\r-----\r$text"
  873.     }
  874.     select [nextLineStart [nextLineStart [minPos]]] [nextLineStart [nextLineStart [nextLineStart [minPos]]]]
  875.     winReadOnly
  876.     message ""
  877. }
  878.  
  879.  
  880. proc getMsgIDs {text} {
  881.     if {[regexp -indices {seld:([0-9]+)} $text dummy ind]} {
  882.         return [concat [string range $text [lindex $ind 0] [lindex $ind 1]] [getMsgIDs [string range $text [lindex $ind 1] end]]]
  883.     }
  884. }
  885.  
  886. proc startNotifying {} {
  887.     global HOME ALPHA quitHooks
  888.     
  889.     AEBuild 'CSOm' CSOm nIns ---- [makeAlis [file join $HOME $ALPHA]]
  890. }
  891.  
  892. proc stopNotifying {} {
  893.     global HOME ALPHA
  894.     
  895.     AEBuild 'CSOm' CSOm nRem ---- [makeAlis [file join $HOME $ALPHA]]
  896. }
  897.  
  898.  
  899.  
  900.  
  901.